'use strict';

const { EventEmitter } = require('events');

let nextItemID = 1;

class TouchBar extends EventEmitter {
  // Bind a touch bar to a window
  static _setOnWindow (touchBar, window) {
    if (window._touchBar != null) {
      window._touchBar._removeFromWindow(window);
    }

    if (touchBar == null) {
      window._setTouchBarItems([]);
      return;
    }

    if (Array.isArray(touchBar)) {
      touchBar = new TouchBar(touchBar);
    }
    touchBar._addToWindow(window);
  }

  constructor (options) {
    super();

    if (options == null) {
      throw new Error('Must specify options object as first argument');
    }

    let { items, escapeItem } = options;

    if (!Array.isArray(items)) {
      items = [];
    }

    this.changeListener = (item) => {
      this.emit('change', item.id, item.type);
    };

    this.windowListeners = {};
    this.items = {};
    this.ordereredItems = [];
    this.escapeItem = escapeItem;

    const registerItem = (item) => {
      this.items[item.id] = item;
      item.on('change', this.changeListener);
      if (item.child instanceof TouchBar) {
        item.child.ordereredItems.forEach(registerItem);
      }
    };

    let hasOtherItemsProxy = false;
    const idSet = new Set();
    items.forEach((item) => {
      if (!(item instanceof TouchBarItem)) {
        throw new Error('Each item must be an instance of TouchBarItem');
      }

      if (item.type === 'other_items_proxy') {
        if (!hasOtherItemsProxy) {
          hasOtherItemsProxy = true;
        } else {
          throw new Error('Must only have one OtherItemsProxy per TouchBar');
        }
      }

      if (!idSet.has(item.id)) {
        idSet.add(item.id);
      } else {
        throw new Error('Cannot add a single instance of TouchBarItem multiple times in a TouchBar');
      }
    });

    // register in separate loop after all items are validated
    for (const item of items) {
      this.ordereredItems.push(item);
      registerItem(item);
    }
  }

  set escapeItem (item) {
    if (item != null && !(item instanceof TouchBarItem)) {
      throw new Error('Escape item must be an instance of TouchBarItem');
    }
    if (this.escapeItem != null) {
      this.escapeItem.removeListener('change', this.changeListener);
    }
    this._escapeItem = item;
    if (this.escapeItem != null) {
      this.escapeItem.on('change', this.changeListener);
    }
    this.emit('escape-item-change', item);
  }

  get escapeItem () {
    return this._escapeItem;
  }

  _addToWindow (window) {
    const { id } = window;

    // Already added to window
    if (Object.prototype.hasOwnProperty.call(this.windowListeners, id)) return;

    window._touchBar = this;

    const changeListener = (itemID) => {
      window._refreshTouchBarItem(itemID);
    };
    this.on('change', changeListener);

    const escapeItemListener = (item) => {
      window._setEscapeTouchBarItem(item != null ? item : {});
    };
    this.on('escape-item-change', escapeItemListener);

    const interactionListener = (event, itemID, details) => {
      let item = this.items[itemID];
      if (item == null && this.escapeItem != null && this.escapeItem.id === itemID) {
        item = this.escapeItem;
      }
      if (item != null && item.onInteraction != null) {
        item.onInteraction(details);
      }
    };
    window.on('-touch-bar-interaction', interactionListener);

    const removeListeners = () => {
      this.removeListener('change', changeListener);
      this.removeListener('escape-item-change', escapeItemListener);
      window.removeListener('-touch-bar-interaction', interactionListener);
      window.removeListener('closed', removeListeners);
      window._touchBar = null;
      delete this.windowListeners[id];
      const unregisterItems = (items) => {
        for (const item of items) {
          item.removeListener('change', this.changeListener);
          if (item.child instanceof TouchBar) {
            unregisterItems(item.child.ordereredItems);
          }
        }
      };
      unregisterItems(this.ordereredItems);
      if (this.escapeItem) {
        this.escapeItem.removeListener('change', this.changeListener);
      }
    };
    window.once('closed', removeListeners);
    this.windowListeners[id] = removeListeners;

    window._setTouchBarItems(this.ordereredItems);
    escapeItemListener(this.escapeItem);
  }

  _removeFromWindow (window) {
    const removeListeners = this.windowListeners[window.id];
    if (removeListeners != null) removeListeners();
  }
}

class TouchBarItem extends EventEmitter {
  constructor () {
    super();
    this._addImmutableProperty('id', `${nextItemID++}`);
    this._parents = [];
  }

  _addImmutableProperty (name, value) {
    Object.defineProperty(this, name, {
      get: function () {
        return value;
      },
      set: function () {
        throw new Error(`Cannot override property ${name}`);
      },
      enumerable: true,
      configurable: false
    });
  }

  _addLiveProperty (name, initialValue) {
    const privateName = `_${name}`;
    this[privateName] = initialValue;
    Object.defineProperty(this, name, {
      get: function () {
        return this[privateName];
      },
      set: function (value) {
        this[privateName] = value;
        this.emit('change', this);
      },
      enumerable: true
    });
  }

  _addParent (item) {
    const existing = this._parents.some(test => test.id === item.id);
    if (!existing) {
      this._parents.push({
        id: item.id,
        type: item.type
      });
    }
  }
}

TouchBar.TouchBarButton = class TouchBarButton extends TouchBarItem {
  constructor (config) {
    super();
    if (config == null) config = {};
    this._addImmutableProperty('type', 'button');
    this._addLiveProperty('label', config.label);
    this._addLiveProperty('accessibilityLabel', config.accessibilityLabel);
    this._addLiveProperty('backgroundColor', config.backgroundColor);
    this._addLiveProperty('icon', config.icon);
    this._addLiveProperty('iconPosition', config.iconPosition);
    this._addLiveProperty('enabled', typeof config.enabled !== 'boolean' ? true : config.enabled);
    if (typeof config.click === 'function') {
      this._addImmutableProperty('onInteraction', () => {
        config.click();
      });
    }
  }
};

TouchBar.TouchBarColorPicker = class TouchBarColorPicker extends TouchBarItem {
  constructor (config) {
    super();
    if (config == null) config = {};
    this._addImmutableProperty('type', 'colorpicker');
    this._addLiveProperty('availableColors', config.availableColors);
    this._addLiveProperty('selectedColor', config.selectedColor);

    if (typeof config.change === 'function') {
      this._addImmutableProperty('onInteraction', (details) => {
        this._selectedColor = details.color;
        config.change(details.color);
      });
    }
  }
};

TouchBar.TouchBarGroup = class TouchBarGroup extends TouchBarItem {
  constructor (config) {
    super();
    if (config == null) config = {};
    this._addImmutableProperty('type', 'group');
    const defaultChild = (config.items instanceof TouchBar) ? config.items : new TouchBar(config.items);
    this._addLiveProperty('child', defaultChild);
    this.child.ordereredItems.forEach((item) => item._addParent(this));
  }
};

TouchBar.TouchBarLabel = class TouchBarLabel extends TouchBarItem {
  constructor (config) {
    super();
    if (config == null) config = {};
    this._addImmutableProperty('type', 'label');
    this._addLiveProperty('label', config.label);
    this._addLiveProperty('accessibilityLabel', config.accessibilityLabel);
    this._addLiveProperty('textColor', config.textColor);
  }
};

TouchBar.TouchBarPopover = class TouchBarPopover extends TouchBarItem {
  constructor (config) {
    super();
    if (config == null) config = {};
    this._addImmutableProperty('type', 'popover');
    this._addLiveProperty('label', config.label);
    this._addLiveProperty('icon', config.icon);
    this._addLiveProperty('showCloseButton', config.showCloseButton);
    const defaultChild = (config.items instanceof TouchBar) ? config.items : new TouchBar(config.items);
    this._addLiveProperty('child', defaultChild);
    this.child.ordereredItems.forEach((item) => item._addParent(this));
  }
};

TouchBar.TouchBarSlider = class TouchBarSlider extends TouchBarItem {
  constructor (config) {
    super();
    if (config == null) config = {};
    this._addImmutableProperty('type', 'slider');
    this._addLiveProperty('label', config.label);
    this._addLiveProperty('minValue', config.minValue);
    this._addLiveProperty('maxValue', config.maxValue);
    this._addLiveProperty('value', config.value);

    if (typeof config.change === 'function') {
      this._addImmutableProperty('onInteraction', (details) => {
        this._value = details.value;
        config.change(details.value);
      });
    }
  }
};

TouchBar.TouchBarSpacer = class TouchBarSpacer extends TouchBarItem {
  constructor (config) {
    super();
    if (config == null) config = {};
    this._addImmutableProperty('type', 'spacer');
    this._addImmutableProperty('size', config.size);
  }
};

TouchBar.TouchBarSegmentedControl = class TouchBarSegmentedControl extends TouchBarItem {
  constructor (config) {
    super();
    if (config == null) config = {};
    this._addImmutableProperty('type', 'segmented_control');
    this._addLiveProperty('segmentStyle', config.segmentStyle);
    this._addLiveProperty('segments', config.segments || []);
    this._addLiveProperty('selectedIndex', config.selectedIndex);
    this._addLiveProperty('mode', config.mode);

    if (typeof config.change === 'function') {
      this._addImmutableProperty('onInteraction', (details) => {
        this._selectedIndex = details.selectedIndex;
        config.change(details.selectedIndex, details.isSelected);
      });
    }
  }
};

TouchBar.TouchBarScrubber = class TouchBarScrubber extends TouchBarItem {
  constructor (config) {
    super();
    if (config == null) config = {};
    let { select, highlight } = config;
    this._addImmutableProperty('type', 'scrubber');
    this._addLiveProperty('items', config.items);
    this._addLiveProperty('selectedStyle', config.selectedStyle || null);
    this._addLiveProperty('overlayStyle', config.overlayStyle || null);
    this._addLiveProperty('showArrowButtons', config.showArrowButtons || false);
    this._addLiveProperty('mode', config.mode || 'free');

    const cont = typeof config.continuous === 'undefined' ? true : config.continuous;
    this._addLiveProperty('continuous', cont);

    if (typeof select === 'function' || typeof highlight === 'function') {
      if (select == null) select = () => {};
      if (highlight == null) highlight = () => {};
      this._addImmutableProperty('onInteraction', (details) => {
        if (details.type === 'select' && typeof select === 'function') {
          select(details.selectedIndex);
        } else if (details.type === 'highlight' && typeof highlight === 'function') {
          highlight(details.highlightedIndex);
        }
      });
    }
  }
};

TouchBar.TouchBarOtherItemsProxy = class TouchBarOtherItemsProxy extends TouchBarItem {
  constructor (config) {
    super();
    this._addImmutableProperty('type', 'other_items_proxy');
  }
};

module.exports = TouchBar;
